﻿<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"
"http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<title>SetTimer</title>
<meta name="description" content="Perform scripted actions at intervals of your choice with this free macro program. SetTimer launches a subroutine automatically and repeatedly.">
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<link href="../css/commands.css" rel="stylesheet" type="text/css">
<link href="../css/print.css" rel="stylesheet" type="text/css" media="print">
</head>
<body>

<h1>SetTimer</h1>

<p>以一个指定的时间间隔来自动重复地启动子程序。</p>

<p class="CommandSyntax">SetTimer, Label [, Period|On|Off, Priority]</p>

<h3>参数 </h3>
<table border="1" width="100%" cellspacing="0" cellpadding="3" bordercolor="#C0C0C0">
  <tr>
    <td width="15%">Label</td>
    <td width="85%">欲跳转的标签或<a href="../Hotkeys.htm">热键标签</a>的名称，它使 <em>Lable</em> 下面的命令被执行直到遇上 <a href="Return.htm">Return </a>或 <a href="Exit.htm">Exit</a>。与几乎所有其他命令的参数一样，<em>Label</em> 可以是一个<a href="../Variables.htm">变量</a>引用比如 %MyLabel%，在这种情况下变量中存储的名称会被作为目标使用。</td>
  </tr>
  <tr>
    <td>Period|On|Off</td>
    <td><p><strong>On</strong>: 重新启用一个在它的先前 <em>period</em> (周期)中被禁用的定时器。如果不存在定时器，则创建它(默认周期为 250)。如果存在定时器但先前被设置成了 <a href="#once">run-only-once 模式</a>，它仍然只运行一次。</p>
      <p><strong>Off</strong>: 禁用一个已存在的定时器。</p>
      <p><strong>Period</strong>: 使用这个参数作为 <em>Label </em>子程序上一次开始后必须经过的<a href="#Precision">近似</a> 毫秒数来创建或更新一个定时器。在这段时间过后，将再次运行 <em>Label </em>（除非它仍在继续上一次运行）。定时器将被自动启用。要防止这种情况，可以随后立即再一次调用此命令，将此参数指定为 OFF。</p>
      <p>如果此参数为空并且:<br>
      1) 定时器不存在：那么将创建一个周期为 250 的定时器。<br>
      2) 定时器已存在：那么将启用定时器，并<a href="#reset">重设</a>它先前的 <em>period</em>，除非指定了 <em>Priority</em>。</p>
      <p><strong><a name="once" id="once"></a>Run only once</strong> [v1.0.46.16+]: 用一个负的 <em>Period </em>来表示定时器只能运行一次。例如，指定 -100 将在 100ms 后运行定时器，然后就像使用了 <em>SetTimer, Label, Off</em> 那样禁用定时器。<br>
      </p>
    </td>
  </tr>
  <tr>
    <td>Priority</td>
    <td><p>这个可选参数是一个介于 -2147483648 和 2147483647 之间的整数（或一个<a href="../Variables.htm#Expressions">表达式</a>），用来表示此定时器的线程优先级。如果省略将视为 0。详见<a href="../misc/Threads.htm">线程</a>。</p>
      <p>要改变一个已有定时器的优先级而不在任何其他方面影响它，只要将此参数前面的那些参数都留空。</p></td>
  </tr>
</table>

<h3>注意</h3>
<p>定时器很有用，因为它们是异步运行的，这意味着即使当脚本正在等待一个窗口，显示一个对话框，或忙于其他任务时，定时器也将按指定的频率（间隔）运行。它们的许多应用实例包括当用户闲置时(如 <a href="../Variables.htm#TimeIdle">%A_TimeIdle%</a> 反映的那样)采取某些行动或者在不需要的窗口出现时关闭它们。</p>
<p>尽管定时器会给人以脚本同时运行多个任务的错觉，但实际并非如此。而是定时的子程序被作为其他线程来处理：它们能中断别的线程或被另一个线程打断，比如<a href="../Hotkeys.htm">热键子程序</a>。详见<a href="../misc/Threads.htm">线程</a>。</p>
<p>无论一个定时器是被创建，重新启用，还是更新为一个新的 <em>period</em>，其子程序都不会马上运行；它的 <em>period </em>时间必须先到期。如果你想让定时器的首次执行马上开始，可用 <a href="Gosub.htm">Gosub</a> 来执行定时器的子程序（不过这种方式不会像定时器自身那样开启一个新线程；所以像 <a href="SendMode.htm">SendMode</a> 这样的设置就不会以它们的默认值启动了）。</p>
<p><a name="reset"></a>如果 SetTimer 被用在一个已存在的定时器上且第二个参数是一个数字或单词 ON(或它被省略)，定时器内部的&ldquo;上次被运行时间&rdquo;将被重置为当前时间；换句话说，必须在定时器经过它的整个周期后，其子程序才会再次运行。</p>
<p><strong><a name="Precision"></a>定时器精度</strong>: 由于操作系统的时间记录系统的间隔原因, <em>周期</em> 通常最接近10或15.6毫秒的倍数(取决于硬件和安装的驱动)。比如，一个在1和10之间(包括10)的<em>周期</em>在Windows NT/2000/XP下通常等于10或15.6(Windows 9x下约为55ms)。通过Loop+Sleep可以获得更短的延时，请看示例：<a href="Sleep.htm#ShorterSleep">DllCall+<span class="NoIndent">timeBeginPeriod</span>+Sleep</a>.</p>
<p>定时器在下列情况中可能无法按指定频率运行：</p>
<ol>
  <li>其它应用程序使 CPU 负担过重。</li>
  <li>定时器的子程序本身花费了比它的周期更长的时间来运行，或有太多相抵触的定时器（调整 <a href="SetBatchLines.htm">SetBatchLines</a> 可能会有所帮助）。</li>
  <li>定时器被另一个<a href="../misc/Threads.htm">线程</a>，即另一个定时的子程序、<a href="../Hotkeys.htm">热键子程序</a>或者<a href="Menu.htm">自定义菜单项</a>打断(这可以通过 <a href="Critical.htm">Critical</a> 避免)。如果发生了这种情况且打断线程花了很长时间才结束，那么实际上被打断的定时器在这段时间中将被禁用。不过，其它的定时器对打断首个定时器的<a href="../misc/Threads.htm">线程</a>进行中断，仍可继续运行。</li>
  <li>脚本由于 <a href="Critical.htm">Critical</a> 或<a href="Thread.htm"> Thread Interrupt/Priority</a> 而不可中断。在此时间内，定时器不会运行。之后，当脚本可再次中断时，延误的定时器会尽快运行一次然后恢复到它的正常运作。</li>
</ol>
<p>由于定时器靠临时中断脚本的当前活动来运作，所以它们的子程序应保持简短（以便它们迅速完成），无论何时都不适宜有很长的中断。</p>
<p>一个脚本运行期间持续生效的定时器通常应在<a href="../Scripts.htm#auto">自动执行部分</a>创建。相比之下，临时定时器经常会被它的子程序禁用（请看本页底部的示例）。</p>
<p>每当一个定时的子程序运行时，它的设置比如 <a href="SendMode.htm">SendMode</a> 都会以默认值重新开始。这些默认值可在<a href="../Scripts.htm#auto">自动执行部分</a>修改。</p>
<p>虽然定时器会在脚本<a href="Suspend.htm">挂起</a>时运作，但如果<a href="../misc/Threads.htm">当前线程</a>的 &quot;<a href="Thread.htm">Thread NoTimers</a>&quot; 正生效或者每当线程被<a href="Pause.htm">暂停</a>时，它们是不会运行的。另外，在用户浏览脚本的某个菜单时（如托盘图标菜单或菜单栏）定时器也不会运作。</p>
<p>如果<a href="../Hotkeys.htm">热键</a>的响应时间很重要（比如游戏中）且脚本包含的定时器子程序执行时间会超过 5 ms，那么可以使用如下命令来避免 15 ms 的延迟。否则如果热键正好在一个定时器线程处于它的不可中断周期时被按下，这个延迟就会发生：</p>
<pre><a href="Thread.htm">Thread</a>, interrupt, 0  <em>;使所有线程始终可中断。</em></pre>
<p>如果定时器在它的子程序正运行时被禁用，该子程序会继续运行直到完成。</p>
<p><a href="KeyHistory.htm">KeyHistory</a> 功能会显示存在多少个定时器，当前启用了多少个。</p>
<p>定时器的周期不可大于 4294967295 毫秒(49.7 天)。</p>
<p>要让脚本持续运行，比如那些只包含了定时器的脚本，请用 <a href="_Persistent.htm">#Persistent</a>。</p>
<h3>相关命令</h3>
<p><a href="Gosub.htm">Gosub</a>, <a href="Return.htm">Return</a>, <a href="../misc/Threads.htm">Threads</a>, <a href="Thread.htm">Thread (command)</a>, <a href="Critical.htm">Critical</a>, <a href="../Functions.htm#IsLabel">IsLabel()</a>, <a href="Menu.htm">Menu</a>, <a href="_Persistent.htm">#Persistent</a></p>
<h3>示例</h3>
<pre class="NoIndent"><em>;例 1: 当不需要的窗口出现时将其关闭：</em>
#Persistent
SetTimer, CloseMailWarnings, 250
return

CloseMailWarnings:
WinClose, Microsoft Outlook, A timeout occured while communicating
WinClose, Microsoft Outlook, A connection to the server could not be established
return</pre>
<p>&nbsp;</p>
<pre class="NoIndent"><em>;例 2: 等待一个特定的窗口出现，然后通知用户：</em>
#Persistent
SetTimer, Alert1, 500
return

Alert1:
IfWinNotExist, Video Conversion, Process Complete
    return
<em>;否则:</em>
SetTimer, Alert1, Off  <em>;即定时器在此处将自己关闭。</em>
SplashTextOn, , , 视频转换已完成。
Sleep, 3000
SplashTextOff
return</pre>
<p>&nbsp;</p>
<pre class="NoIndent"><em>;例 3: 检测热键的单击、双击和三次按击。</em>
<em>;这允许热键可以按你按键的次数来执行不同</em><em>的操作：</em>
#c::
if winc_presses &gt; 0 <em>; SetTimer 已经启动，所以我们记录按键。</em>
{
    winc_presses += 1
    return
}
<em>;否则，这是新一系列按键的首次按键。将计数设为 1 并</em><em>启动定时器：</em>
winc_presses = 1
SetTimer, KeyWinC, 400 <em>;在 400 毫秒内等待更多的按键。</em>
return

KeyWinC:
SetTimer, KeyWinC, off
if winc_presses = 1 <em>;该键已按过一次。</em>
{
    Run, m:\  <em>;打开一个文件夹。</em>
}
else if winc_presses = 2 <em>;该键已按过两次。</em>
{
    Run, m:\multimedia  <em>;打开一个不同的文件夹。</em>
}
else if winc_presses &gt; 2
{
    MsgBox, 检测到三次或更多次点击。
}
<em>;不论上面哪个动作被触发，将计数复位以备下一系列的按键：</em><em></em>
winc_presses = 0
return</pre>

</body>
</html>
